1   /**
2    * Copyright (C) 2006-2019 INRIA and contributors
3    *
4    * Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) of the Cecill-C License (see LICENSE-CECILL-C.txt). You as the user are entitled to choose the terms under which to adopt Spoon.
5    */
6   package spoon.reflect.visitor;
7   
8   import spoon.processing.AbstractProcessor;
9   import spoon.processing.Processor;
10  import spoon.reflect.code.CtConstructorCall;
11  import spoon.reflect.code.CtTargetedExpression;
12  import spoon.reflect.declaration.CtCompilationUnit;
13  import spoon.reflect.declaration.CtElement;
14  import spoon.reflect.declaration.CtPackage;
15  import spoon.reflect.path.CtRole;
16  import spoon.reflect.reference.CtExecutableReference;
17  import spoon.reflect.reference.CtFieldReference;
18  import spoon.reflect.reference.CtTypeReference;
19  import spoon.reflect.reference.CtVariableReference;
20  import spoon.reflect.visitor.chain.CtScannerListener;
21  import spoon.reflect.visitor.chain.ScanningMode;
22  import spoon.support.Experimental;
23  
24  import java.util.Arrays;
25  import java.util.HashSet;
26  import java.util.Set;
27  
28  /**
29   *{@link Processor} of {@link CtCompilationUnit}, which scans CtCompilationUnit modules, packages and types
30   * with purpose to find type references and expressions which might influence import directives.
31   *
32   * Subclasses create a scanner ({@link #createScanner()}) and analyzes the elements to be imported {@link #handleTypeReference} and {@link #handleTargetedExpression(CtTargetedExpression, Object)}
33   *
34   */
35  @Experimental
36  abstract class ImportAnalyzer<U> extends AbstractProcessor<CtElement> {
37  
38  	protected EarlyTerminatingScanner scanner;
39  	@Override
40  	public void process(CtElement el) {
41  		scanner = createScanner();
42  		CtScannerListener listener = createScannerListener();
43  		scanner.setListener(listener);
44  		if (el instanceof CtCompilationUnit) {
45  			process(scanner, (CtCompilationUnit) el);
46  		} else {
47  			scanner.scan(el);
48  		}
49  	}
50  
51  	protected static void process(CtScanner scanner, CtCompilationUnit cu) {
52  		scanner.enter(cu);
53  		switch (cu.getUnitType()) {
54  		case MODULE_DECLARATION:
55  		case UNKNOWN:
56  			break;
57  		case PACKAGE_DECLARATION:
58  			// we need to compute imports only for package annotations and package comments
59  			// we don't want to get all imports coming from content of package
60  			CtPackage pack = cu.getDeclaredPackage();
61  			scanner.scan(pack.getAnnotations());
62  			break;
63  		case TYPE_DECLARATION:
64  			for (CtTypeReference<?> typeRef : cu.getDeclaredTypeReferences()) {
65  				scanner.scan(typeRef.getTypeDeclaration());
66  			}
67  			break;
68  		}
69  		scanner.exit(cu);
70  	}
71  
72  	protected CtScannerListener createScannerListener() {
73  		return new ScannerListener();
74  	}
75  
76  	//The set of roles whose values are always kept implicit
77  	protected static Set<CtRole> IGNORED_ROLES_WHEN_IMPLICIT = new HashSet<>(Arrays.asList(
78  			//e.g. List<String> s = new ArrayList</*keep me implicit*/>();
79  			CtRole.TYPE_ARGUMENT,
80  			//e.g. List<?/* extends Object*/>
81  			CtRole.BOUNDING_TYPE,
82  			//e.g. (/*implicit type of parameter*/ p) -> {}
83  			CtRole.TYPE
84  	));
85  
86  	/**
87  	 * {@link CtScannerListener} implementation which stops scanning of children on elements,
88  	 * which mustn't have influence to compilation unit imports.
89  	 */
90  	protected class ScannerListener implements CtScannerListener {
91  		protected Set<CtRole> ignoredRoles = IGNORED_ROLES_WHEN_IMPLICIT;
92  
93  		ScannerListener() {
94  			super();
95  		}
96  
97  		@Override
98  		public ScanningMode enter(CtRole role, CtElement element) {
99  			if (element == null) {
100 				return ScanningMode.SKIP_ALL;
101 			}
102 			if (role == CtRole.VARIABLE && element instanceof CtVariableReference) {
103 				//ignore variable reference of field access. The accessType is relevant here instead.
104 				return ScanningMode.SKIP_ALL;
105 			}
106 			if (element.isParentInitialized()) {
107 				CtElement parent = element.getParent();
108 				if (role == CtRole.DECLARING_TYPE && element instanceof CtTypeReference) {
109 					if (parent instanceof CtFieldReference) {
110 						//ignore the declaring type of field reference. It is not relevant for Imports
111 						return ScanningMode.SKIP_ALL;
112 					}
113 					if (parent instanceof CtExecutableReference) {
114 						/*
115 						 * ignore the declaring type of type executable like
116 						 * anVariable.getSomeInstance().callMethod()
117 						 * The declaring type of `callMethod` method is not relevant for Imports
118 						 */
119 						return ScanningMode.SKIP_ALL;
120 					} else if (parent instanceof CtTypeReference) {
121 						/*
122 						 * It looks like this is not needed too.
123 						 *
124 						 * pvojtechovsky: I am sure it is not wanted in case of
125 						 * spoon.test.imports.testclasses.internal.ChildClass.InnerClassProtected
126 						 * which extends package protected (and for others invisible class)
127 						 * spoon.test.imports.testclasses.internal.SuperClass
128 						 * and in this case the import directive must import ...ChildClass and not ...SuperClass,
129 						 * because import is using type "access path" and not qualified name of the type.
130 						 *
131 						 * ... but in other normal cases, I guess the declaring type is used and needed for import!
132 						 * ... so I don't understand why SKIP_ALL works in all cases. May be there is missing test case?
133 						 */
134 						if (!((CtTypeReference) parent).getAccessType().equals(element)) {
135 							return ScanningMode.SKIP_ALL;
136 						}
137 					}
138 				}
139 				if (role == CtRole.TYPE && element instanceof CtTypeReference) {
140 					if (parent instanceof CtFieldReference) {
141 						//ignore the type of field references. It is not relevant for Imports
142 						return ScanningMode.SKIP_ALL;
143 					}
144 					if (parent instanceof CtExecutableReference) {
145 						CtElement parent2 = null;
146 						if (parent.isParentInitialized()) {
147 							parent2 = parent.getParent();
148 						}
149 						if (parent2 instanceof CtConstructorCall<?>) {
150 							//new SomeType(); is relevant for import
151 							//continue
152 						} else {
153 							/*
154 							 * ignore the return type of executable reference. It is not relevant for Imports
155 							 * anVariable.getSomeInstance().callMethod()
156 							 * The return type `callMethod` method is not relevant for Imports
157 							 */
158 							return ScanningMode.SKIP_ALL;
159 						}
160 					}
161 					/*
162 					 * CtTypeReference
163 					 * CtMethod, CtField, ...
164 					 * continue. This is relevant for import
165 					 */
166 				}
167 				if (role == CtRole.ARGUMENT_TYPE) {
168 					/*
169 					 * ignore the type of parameter of CtExecutableReference
170 					 * It is not relevant for Imports.
171 					 */
172 					return ScanningMode.SKIP_ALL;
173 				}
174 			}
175 			if (element.isImplicit() && ignoredRoles.contains(role)) {
176 				//ignore implicit actual type arguments
177 				return ScanningMode.SKIP_ALL;
178 			}
179 			onEnter(getScannerContextInformation(), role, element);
180 			return ScanningMode.NORMAL;
181 		}
182 	}
183 
184 
185 	protected void onEnter(U context, CtRole role, CtElement element) {
186 
187 		if (element instanceof CtTargetedExpression) {
188 			CtTargetedExpression<?, ?> targetedExpression = (CtTargetedExpression<?, ?>) element;
189 			handleTargetedExpression(targetedExpression, context);
190 		} else if (element instanceof CtTypeReference<?>) {
191 			//we have to visit only PURE CtTypeReference. No CtArrayTypeReference, CtTypeParameterReference, ...
192 			element.accept(new CtAbstractVisitor() {
193 				@Override
194 				public <T> void visitCtTypeReference(CtTypeReference<T> reference) {
195 					handleTypeReference((CtTypeReference<?>) element, context, role);
196 				}
197 			});
198 		}
199 	}
200 
201 	/** extract the required information from the scanner to take a decision */
202 	protected abstract U getScannerContextInformation();
203 
204 	/** creates the scanner that will be used to visit the model */
205 	protected abstract EarlyTerminatingScanner createScanner();
206 
207 	/** what do we do a type reference? */
208 	protected abstract void handleTypeReference(CtTypeReference<?> element, U context, CtRole role);
209 
210 	/** what do we do a target expression (print target or not) ? */
211 	protected abstract void handleTargetedExpression(CtTargetedExpression<?, ?> targetedExpression, U context);
212 
213 	/**
214 	 * @return parent of `element`, but only if it's type is `type`
215 	 */
216 	protected static <T extends CtElement> T getParentIfType(CtElement element, Class<T> type) {
217 		if (element == null || !element.isParentInitialized()) {
218 			return null;
219 		}
220 		CtElement parent = element.getParent();
221 		if (type.isInstance(parent)) {
222 			return type.cast(parent);
223 		}
224 		return null;
225 	}
226 }